homework 8, version 1
Submission by: Jazzy Doe (jazz@mit.edu)
Bonus homework 8: Raytracing in 3D
18.S191
, fall 2020
This week we will recreate the 3D raytracer from the lectures, building on what we wrote for the 2D case. Compared to the past weeks, you will notice that this homework is a little different! We have included fewer intermediate checks, it is up to you to decide how you:
split up the problem is smaller functions
check and visualize your progress
To guide you through the homework, you can follow along with this week's live coding session. Feel free to stay close to James's code, or to come up with your own solution.
With the raytracer done, we will be able to recreate an artwork by M.C. Escher, scroll down to have a sneak peek!
For MIT students: this homework is optional.
Feel free to ask questions!
"Jazzy Doe"
"jazz"
Lecture
Live coding session
Let's create a package environment:
Activating new project at `/tmp/jl_ChwFH2` Updating registry at `~/.julia/registries/General.toml` Resolving package versions... Updating `/tmp/jl_ChwFH2/Project.toml` [6218d12a] + ImageMagick v1.3.0 [916415d5] + Images v0.23.3 [91a5bcdd] + Plots v1.40.7 [7f904dfe] + PlutoUI v0.6.11 Updating `/tmp/jl_ChwFH2/Manifest.toml` [621f4979] + AbstractFFTs v1.5.0 [79e6a3ab] + Adapt v4.2.0 [13072b0f] + AxisAlgorithms v1.0.1 [39de3d68] + AxisArrays v0.4.7 [d1d4a3ce] + BitFlags v0.1.9 [aafaddc9] + CatIndices v0.2.2 [d360d2e6] + ChainRulesCore v1.25.1 [9e997f8a] + ChangesOfVariables v0.1.9 [944b1d66] + CodecZlib v0.7.8 [35d6a980] + ColorSchemes v3.26.0 [3da002f7] + ColorTypes v0.10.12 [c3611d14] + ColorVectorSpace v0.8.7 [5ae59095] + Colors v0.12.11 [34da2185] + Compat v4.16.0 [ed09eef8] + ComputationalResources v0.3.2 [f0e56b4a] + ConcurrentUtilities v2.5.0 [187b0558] + ConstructionBase v1.5.8 [d38c429a] + Contour v0.6.3 [150eb455] + CoordinateTransformations v0.6.3 [dc8bdbbb] + CustomUnitRanges v1.0.2 [9a962f9c] + DataAPI v1.16.0 [864edb3b] + DataStructures v0.18.20 [b4f34e82] + Distances v0.10.12 [ffbed154] + DocStringExtensions v0.9.3 [460bff9d] + ExceptionUnwrapping v0.1.11 [c87230d0] + FFMPEG v0.4.2 [4f61f5a4] + FFTViews v0.3.2 [7a1cc6ca] + FFTW v1.8.1 [5789e2e9] + FileIO v1.17.0 [53c48c17] + FixedPointNumbers v0.8.5 [1fa38f19] + Format v1.3.7 [28b8d3ca] + GR v0.72.8 [a2bd30eb] + Graphics v1.1.3 [42e2da0e] + Grisu v1.0.2 [cd3eb016] + HTTP v1.10.15 [bbac6d45] + IdentityRanges v0.3.1 [2803e5a7] + ImageAxes v0.6.9 [f332f351] + ImageContrastAdjustment v0.3.7 [a09fc81d] + ImageCore v0.8.22 [51556ac3] + ImageDistances v0.2.13 [6a3955dd] + ImageFiltering v0.6.21 [6218d12a] + ImageMagick v1.3.0 [bc367c6b] + ImageMetadata v0.9.5 [787d08f9] + ImageMorphology v0.2.11 [2996bd0c] + ImageQualityIndexes v0.2.2 [4e3cecfd] + ImageShow v0.2.3 [02fcd773] + ImageTransformations v0.8.13 [916415d5] + Images v0.23.3 [9b13fd28] + IndirectArrays v0.5.1 [a98d9a8b] + Interpolations v0.13.6 [8197267c] + IntervalSets v0.7.10 [3587e190] + InverseFunctions v0.1.17 [92d709cd] + IrrationalConstants v0.1.1 [c8e1da08] + IterTools v1.4.0 [1019f520] + JLFzf v0.1.9 [692b3bcd] + JLLWrappers v1.7.0 [682c06a0] + JSON v0.21.4 [b964fa9f] + LaTeXStrings v1.4.0 [23fbe1c1] + Latexify v0.16.6 [2ab3a3ac] + LogExpFunctions v0.3.28 [e6f89c97] + LoggingExtras v1.1.0 [1914dd2f] + MacroTools v0.5.15 [dbb5928d] + MappedArrays v0.4.2 [739be429] + MbedTLS v1.1.9 [442fdcdd] + Measures v0.3.2 [e1d29d7a] + Missings v1.2.0 [e94cdb99] + MosaicViews v0.3.4 [77ba4419] + NaNMath v1.0.3 [6fe1bfb0] + OffsetArrays v1.15.0 [4d8831e6] + OpenSSL v1.4.3 [bac558e1] + OrderedCollections v1.8.0 [5432bcbf] + PaddedViews v0.5.12 [d96e819e] + Parameters v0.12.3 [69de0a69] + Parsers v2.8.1 [b98c9c47] + Pipe v1.3.0 [ccf2f8ad] + PlotThemes v3.3.0 [995b91a9] + PlotUtils v1.4.3 [91a5bcdd] + Plots v1.40.7 [7f904dfe] + PlutoUI v0.6.11 [aea7be01] + PrecompileTools v1.2.1 [21216c6a] + Preferences v1.4.3 [94ee1d12] + Quaternions v0.7.6 [b3c3ace0] + RangeArrays v0.3.2 [c84ed2f1] + Ratios v0.4.5 [c1ae055f] + RealDot v0.1.0 [3cdcf5f2] + RecipesBase v1.3.4 [01d81517] + RecipesPipeline v0.6.12 [189a3867] + Reexport v1.2.2 [05181044] + RelocatableFolders v1.0.1 [ae029012] + Requires v1.3.1 [6038ab10] + Rotations v1.7.1 [6c6a2e73] + Scratch v1.2.1 [992d4aef] + Showoff v1.0.3 [777ac1f9] + SimpleBufferStream v1.2.0 [699a6c99] + SimpleTraits v0.9.4 [a2af1166] + SortingAlgorithms v1.2.1 [276daf66] + SpecialFunctions v1.8.8 [860ef19b] + StableRNGs v1.0.2 [cae243ae] + StackViews v0.1.1 [90137ffa] + StaticArrays v1.9.13 [1e83bf80] + StaticArraysCore v1.4.3 [82ae8749] + StatsAPI v1.7.0 [2913bbd2] + StatsBase v0.33.21 [fd094767] + Suppressor v0.2.8 [06e1c1a7] + TiledIteration v0.3.1 [3bb67fe8] + TranscodingStreams v0.11.3 [5c2747f8] + URIs v1.5.1 [3a884ed6] + UnPack v1.0.2 [1cfade01] + UnicodeFun v0.4.1 [1986cc42] + Unitful v1.22.0 [45397f5d] + UnitfulLatexify v1.6.4 [41fe7b60] + Unzip v0.2.0 [efce3f68] + WoodburyMatrices v0.5.6 [6e34b625] + Bzip2_jll v1.0.9+0 [83423d85] + Cairo_jll v1.18.2+1 [ee1fde0b] + Dbus_jll v1.14.10+0 [2702e6a9] + EpollShim_jll v0.0.20230411+1 [2e619515] + Expat_jll v2.6.5+0 [b22a6f82] + FFMPEG_jll v4.4.2+2 [f5851436] + FFTW_jll v3.3.10+3 [a3f928ae] + Fontconfig_jll v2.15.0+0 [d7e528f0] + FreeType2_jll v2.13.3+1 [559328eb] + FriBidi_jll v1.0.16+0 [0656b61e] + GLFW_jll v3.4.0+2 [d2c73de3] + GR_jll v0.72.8+0 [78b55507] + Gettext_jll v0.21.0+0 [7746bdde] + Glib_jll v2.82.4+0 [3b182d85] + Graphite2_jll v1.3.14+1 [2e76f6c2] + HarfBuzz_jll v8.5.0+0 [c73af94c] + ImageMagick_jll v6.9.10-12+3 [1d5cc7b8] + IntelOpenMP_jll v2025.0.4+0 [aacddb02] + JpegTurbo_jll v3.1.1+0 [c1c5ebd0] + LAME_jll v3.100.2+0 [88015f11] + LERC_jll v3.0.0+1 [1d63c593] + LLVMOpenMP_jll v18.1.7+0 [dd4b983a] + LZO_jll v2.10.3+0 [e9f186c6] + Libffi_jll v3.2.2+2 [d4300ac3] + Libgcrypt_jll v1.11.0+0 [7e76a0d4] + Libglvnd_jll v1.7.0+0 [7add5ba3] + Libgpg_error_jll v1.51.1+0 [94ce4f54] + Libiconv_jll v1.18.0+0 [4b2f31a3] + Libmount_jll v2.40.3+0 [89763e89] + Libtiff_jll v4.4.0+0 [38a345b3] + Libuuid_jll v2.40.3+0 [856f044c] + MKL_jll v2025.0.1+1 [e7412a2a] + Ogg_jll v1.3.5+1 [458c3c95] + OpenSSL_jll v1.1.23+1 [efe28fd5] + OpenSpecFun_jll v0.5.6+0 [91d4177d] + Opus_jll v1.3.3+0 [36c8627f] + Pango_jll v1.56.1+0 [30392449] + Pixman_jll v0.43.4+0 [ea2cea3b] + Qt5Base_jll v5.15.3+2 [a2964d1f] + Wayland_jll v1.21.0+2 [2381bf8a] + Wayland_protocols_jll v1.36.0+0 [02c8fc9c] + XML2_jll v2.13.6+1 [aed1982a] + XSLT_jll v1.1.42+0 [4f6342f7] + Xorg_libX11_jll v1.8.6+3 [0c0b7dd1] + Xorg_libXau_jll v1.0.12+0 [935fb764] + Xorg_libXcursor_jll v1.2.3+0 [a3789734] + Xorg_libXdmcp_jll v1.1.5+0 [1082639a] + Xorg_libXext_jll v1.3.6+3 [d091e8ba] + Xorg_libXfixes_jll v6.0.0+0 [a51aa0fd] + Xorg_libXi_jll v1.8.2+0 [d1454406] + Xorg_libXinerama_jll v1.1.5+0 [ec84b674] + Xorg_libXrandr_jll v1.5.4+0 [ea2f1a96] + Xorg_libXrender_jll v0.9.11+1 [14d82f49] + Xorg_libpthread_stubs_jll v0.1.2+0 [c7cfdc94] + Xorg_libxcb_jll v1.17.0+3 [cc61e674] + Xorg_libxkbfile_jll v1.1.2+1 [12413925] + Xorg_xcb_util_image_jll v0.4.0+1 [2def613f] + Xorg_xcb_util_jll v0.4.0+1 [975044d2] + Xorg_xcb_util_keysyms_jll v0.4.0+1 [0d47668e] + Xorg_xcb_util_renderutil_jll v0.3.9+1 [c22f9ab0] + Xorg_xcb_util_wm_jll v0.4.1+1 [35661453] + Xorg_xkbcomp_jll v1.4.6+1 [33bec58e] + Xorg_xkeyboard_config_jll v2.39.0+0 [c5fb5394] + Xorg_xtrans_jll v1.5.1+0 [3161d3a3] + Zstd_jll v1.5.7+1 [214eeab7] + fzf_jll v0.56.3+0 [a4ae2306] + libaom_jll v3.11.0+0 [0ac62f75] + libass_jll v0.15.2+0 [1183f4f0] + libdecor_jll v0.2.2+0 [f638f0a6] + libfdk_aac_jll v2.0.3+0 [b53b4c65] + libpng_jll v1.6.47+0 [f27f6e37] + libvorbis_jll v1.3.7+2 [1317d2d5] + oneTBB_jll v2022.0.0+0 [1270edf5] + x264_jll v2021.5.5+0 [dfaa095f] + x265_jll v3.5.0+0 [d8fb68d0] + xkbcommon_jll v1.4.1+2 [0dad84c5] + ArgTools [56f22d72] + Artifacts [2a0f44e3] + Base64 [ade2ca70] + Dates [8bb1440f] + DelimitedFiles [8ba89e20] + Distributed [f43a241f] + Downloads [7b1f6079] + FileWatching [b77e0a4c] + InteractiveUtils [4af54fe1] + LazyArtifacts [b27032c2] + LibCURL [76f85450] + LibGit2 [8f399da3] + Libdl [37e2e46d] + LinearAlgebra [56ddb016] + Logging [d6f4376e] + Markdown [a63ad114] + Mmap [ca575930] + NetworkOptions [44cfe95a] + Pkg [de0858da] + Printf [3fa0cd96] + REPL [9a3f8284] + Random [ea8e919c] + SHA [9e88b42a] + Serialization [1a1011a3] + SharedArrays [6462fe0b] + Sockets [2f01184e] + SparseArrays [10745b16] + Statistics [fa267f1f] + TOML [a4e569a6] + Tar [8dfed614] + Test [cf7118a7] + UUIDs [4ec0a83e] + Unicode [e66e0078] + CompilerSupportLibraries_jll [deac9b47] + LibCURL_jll [29816b5a] + LibSSH2_jll [c8ffd9c3] + MbedTLS_jll [14a3606d] + MozillaCACerts_jll [4536629a] + OpenBLAS_jll [05823500] + OpenLibm_jll [efcefdf7] + PCRE2_jll [83775a58] + Zlib_jll [8e850b90] + libblastrampoline_jll [8e850ede] + nghttp2_jll [3f19e933] + p7zip_jll
From the last homework
Below we have included some important functions from the last homework (Raytracing in 2D), which we will be able to re-use for the 3D case.
There are some small changes:
The concept of a
Photon
now carries color information.The
Sphere
is no longer a pure lens, it contains aSurface
property which describes the mixture between transmission, reflection and a pure color. More on this later!The
refract
function is updated to handle two edge cases, but its behaviour is generally unchanged.
Outside of these changes, all functions from the previous homework can be taken "as-is" when converting to 3D, cool!
Intersections:
closest_hit (generic function with 1 method)
Reflect and refract:
reflect (generic function with 1 method)
refract (generic function with 1 method)
Surface (new)
Sphere
Aasdf
intersection (generic function with 1 method)
sphere_normal_at (generic function with 1 method)
Camera and Skyboxes
Now we can begin looking into the 3D nature of raytracing to create visualizations similar to those in lecture. The first step is setting up the camera and another stuct known as a sky box to collect all the rays of light.
Luckily, the transition from 2D to 3D for raytracing is relatively straightforward and we can use all of the functions and concepts we have built in 2D moving forward.
Firstly, the camera:
Camera
For the purposes of this homework, we will constrain ourselves to a camera pointing exclusively downward. This is simply because camera positioning can be a bit tricky and there is no reason to make the homework more complicated than it needs to be!
So, what is the purpose of the camera?
Well, in reality, a camera is a device that collects the color information from all the rays of light that are refracting and reflecting off of various objects in some sort of scene. Because there are a nearly infinite number of rays bouncing around the scene at any time, we will actually constrain ourselves only to rays that are entering our camera. In poarticular, we will create a 2D screen just in front of the camera and send a ray from the camera to each pixel in the screen, as shown in the following image:

Because we are not considering camera motion for this exercise, we will assume that the image plane is constrained to the horizontal plane, but that the camera, itself, can be some distance behind it. This distance from the image plane to the camera is called the focal length and is used to determine the field of view.
From here, it's clear we need to construct:
A camera struct
A function to initialize all the rays being generated by the camera
Let's start with the struct
400
300
9.0
-10.0
0.0
0.0
100.0
Now we need to construct some method to create each individual ray extending from the camera to a pixel in the image plane.
init_rays (generic function with 1 method)
4
3
16.0
-5.0
0.0
20.0
100.0
3×4 Matrix{Photon}:
Photon([0.0, 20.0, 100.0], [-0.715542, 0.536656, -0.447214], RGB{N0f8}(0.0,0.0,0.0), 1.0) … Photon([0.0, 20.0, 100.0], [0.715542, 0.536656, -0.447214], RGB{N0f8}(0.0,0.0,0.0), 1.0)
Photon([0.0, 20.0, 100.0], [-0.847998, 0.0, -0.529999], RGB{N0f8}(0.0,0.0,0.0), 1.0) Photon([0.0, 20.0, 100.0], [0.847998, 0.0, -0.529999], RGB{N0f8}(0.0,0.0,0.0), 1.0)
Photon([0.0, 20.0, 100.0], [-0.715542, -0.536656, -0.447214], RGB{N0f8}(0.0,0.0,0.0), 1.0) Photon([0.0, 20.0, 100.0], [0.715542, -0.536656, -0.447214], RGB{N0f8}(0.0,0.0,0.0), 1.0)
extract_colors (generic function with 1 method)
Nothing yet... time to add some objects!
Skybox
Now that we have the concept of a camera, we can technically do a fully 3D raytracing example; however, we want to ensure that each pixel will actually hit something – preferrably something with some color gradient so we can make sure our simulation is working!
For this, we will introduce the concept of a sky box, which is standard for most gaming applications. Here, the idea is that our entire scene is held within some additional object, just like the mirrors we used in the 2D example. The only difference here is that we will be using some texture instead of a reflective surface. In addition, even though we are calling it a box, we'll actually be treating it as a sphere.
Because we have already worked out how to make sure we have hit the interior of a spherical lens, we will be using a similar function here. For this part of the exercise, we will need to construct 2 things:
A skybox struct
A function that returns some color gradient to be called whenever a ray of light interacts with a sky box
So let's start with the sky box struct
Now we have the ability to create a skybox, the only thing left is to create some sort of texture function so that when the ray of light hits the sky box, we can return some form of color information. So for this, we will basically create a function that returns back a smooth gradient in different directions depending on the position of the ray when it hits the skybox.
For the color information, we will be assigning a color to each cardinal axis. That is to say that there will be a red gradient along
where
So let's get to it and write the function!
gradient_skybox_color (generic function with 1 method)
0.0
0.0
0.0
1000.0
gradient_skybox_color (generic function with 1 method)
Let's set up a basic scene and trace an image! Since our skybox is spherical we can use the same intersect
method as we use for Sphere
s. Have a look at the intersect
method, we already added SkyBox
as a possible type.
300
200
16.0
-5.0
0.0
20.0
100.0
To create this image, we used the ray tracing function bewlow, which takes in a camera and a set of objects / scene, and...
Initilializes all the rays
Propagates the rays forward
Converts everything into an image
ray_trace (generic function with 1 method)
Writing a ray tracer
It's your turn! Below is the code needed to trace just the sky box, but we still need to add the ability to trace spheres.
We recommend that you start by just implementing reflection - make every sphere reflect, regardless of its surface. Make sure that this is working well - can you see the reflection of one sphere in another sphere? Does our program get stuck in a loop?
Once you have reflections working, you can add refraction and colored spheres. In the 2D example, we dealt specifically with spheres that could either 100% reflect or refract. In reality, it is possible for objects to either reflect or refract, something in-between. That is to say, a ray of light can split when hitting an object surface, creating at least 2 more rays of light that will both return separate color values. The color that we perceive for that ray is the combination both of these colors - they are mixed.
A third possibility explored in the lecture is that the objects can also have a color associated with them and just return the color value instead of reflecting or refracting.
You can choose! After implementing reflection, you can implement three different spheres (you can modify the existing code, create new types, add functions, and so on), a purely reflective, purely refractive or opaquely colored sphere. You can also go straight for the more photorealistic option, which is that every sphere is a combination of these three - this is what we did in the lecture.
interact (generic function with 1 method)
interact (generic function with 2 methods)
step_ray (generic function with 1 method)
Below, we create a scene with a number of balls inside of it. While working on your code, work in small increments, and do frequent checks to see if your code is working. Feel free to modify this test scene, or to create a simpler one.
0.0
0.0
0.0
1000.0
gradient_skybox_color
0.0
0.0
-25.0
20.0
1.0
0.0
1.5
0.0
50.0
-100.0
20.0
0.0
1.0
1.0
-50.0
0.0
-25.0
20.0
0.0
0.0
1.0
30.0
25.0
-60.0
20.0
0.0
0.75
1.5
50.0
0.0
-25.0
20.0
0.5
0.0
1.5
-30.0
25.0
-60.0
20.0
0.5
0.5
1.5
Hint
If you are getting a "Circular Defintions" error - this could be because of a Pluto limitation. If two functions call each other, they need to be contained in a single cell, using a begin end
block.
Bonus: Escher
If you managed to get through the exercises, we have a fun bonus exercise! The goal is to recreate this self-portrait by M.C. Escher:

It looks like M.C. Escher is a skillful raytracer, but so are we! To recreate this image, we can simplify it by having just two objects in our scene:
A purely reflective sphere
A skybox, containing an image of us!
Let's start with our old skybox, and set up our scene:
0.0
0.0
0.0
20.0
1.0
0.0
1.5
300
300
30.0
-10.0
0.0
0.0
30.0
👆 You can modify escher_cam
to increase or descrease the resolution!
Awesome! Next, we want to set an image as our skybox, instead of a gradient. To do so, we have written a function that converts the x,y,z coordinates of the intersection point with a skybox into a latitude/longitude pair, which we can use (after scaling, rounding & clamping) as pixel coordinates to index an image!
image_skybox (generic function with 1 method)
get_index_rational (generic function with 1 method)
0.0
0.0
0.0
1000.0
#7 (generic function with 1 method)
Great! It's like the Earth, but distorted. Notice that the continents are mirrored, and that you can see both poles at the same time.
Okay, self portrait time! Let's take a picture using your webcam, and we will use it as the skybox texture:
MethodError: no method matching getindex(::Missing, ::String)
Here is what happened, the most recent locations are first:
- process_raw_camera_data
(raw_camera_data::Missing) from Other cell: line 13# So to get the red values for each pixel, we take every 4th value, starting at
# the 1st:
reds_flat = UInt8.(raw_camera_data["data"][1:4:end])
greens_flat = UInt8.(raw_camera_data["data"][2:4:end])
blues_flat = UInt8.(raw_camera_data["data"][3:4:end])
- Show more...
Another cell defining face contains errors.
👀 wow! It's Planet Jazzy Doe, surrounded by even more Jazzy Doe.
When you look at the drawing by Escher, you see that he only occupies a small section of the 'skybox'. Behind Escher, you can see his cozy house, and in front of him (i.e. behind the glass sphere, from his persective), you see a gray background.
What we need is a 360° panoramic image of our room. One option is that you make one! There are great tutorials online, and maybe you can use an app to do this with your phone.
Another option is that we approximate the panaroma by padding the image of your face. You can pad the image with a solid color, you can use a gradient, you can use the earth satellite image, you can add noise, and so on. Here is an example of what I made.
padded (generic function with 1 method)
Another cell defining face contains errors.
Let's put it all together!
MethodError: no method matching getindex(::Missing, ::String)
Here is what happened, the most recent locations are first:
- process_raw_camera_data
(raw_camera_data::Missing) from Other cell: line 13# So to get the red values for each pixel, we take every 4th value, starting at
# the 1st:
reds_flat = UInt8.(raw_camera_data["data"][1:4:end])
greens_flat = UInt8.(raw_camera_data["data"][2:4:end])
blues_flat = UInt8.(raw_camera_data["data"][3:4:end])
- Show more...
Before you submit
Remember to fill in your name and Kerberos ID at the top of this notebook.
Function library
Just some helper functions used in the notebook.
process_raw_camera_data (generic function with 1 method)
camera_input (generic function with 1 method)
hint (generic function with 1 method)
almost (generic function with 1 method)
still_missing (generic function with 2 methods)
keep_working (generic function with 2 methods)
Fantastic!
Splendid!
Great!
Yay ❤
Great! 🎉
Well done!
Keep it up!
Good job!
Awesome!
You got the right answer!
Let's move on to the next section.
correct (generic function with 2 methods)
not_defined (generic function with 1 method)
TODO_note (generic function with 1 method)